home *** CD-ROM | disk | FTP | other *** search
/ Internet Info 1994 March / Internet Info CD-ROM (Walnut Creek) (March 1994).iso / networking / ip / ka9q / MNetsrc.hqx / Mac TCP_IP Source v.33 / tcpcmd.c < prev    next >
C/C++ Source or Header  |  1989-04-19  |  6KB  |  305 lines

  1. #include <stdio.h>
  2. #include "global.h"
  3. #include "timer.h"
  4. #include "mbuf.h"
  5. #include "netuser.h"
  6. #include "internet.h"
  7. #include "tcp.h"
  8. #include "cmdparse.h"
  9.  
  10. /* TCP connection states */
  11. char *tcpstates[] = {
  12.     "Closed",
  13.     "Listen",
  14.     "SYN sent",
  15.     "SYN received",
  16.     "Established",
  17.     "FIN wait 1",
  18.     "FIN wait 2",
  19.     "Close wait",
  20.     "Closing",
  21.     "Last ACK",
  22.     "Time wait"
  23. };
  24.  
  25. /* TCP closing reasons */
  26. char *reasons[] = {
  27.     "Normal",
  28.     "Reset",
  29.     "Timeout",
  30.     "ICMP"
  31. };
  32. /* TCP subcommand table */
  33. int domss(),doirtt(),dortt(),dotcpstat(),dowindow(),dotcpkick(),dotcpreset();
  34. struct cmds tcpcmds[] = {
  35.     "irtt",        doirtt,        0,    NULLCHAR,    NULLCHAR,
  36.     "kick",        dotcpkick,    2,    "tcp kick <tcb>",
  37.         NULLCHAR,
  38.     "mss",        domss,        0,    NULLCHAR,    NULLCHAR,
  39.     "reset",    dotcpreset,    2,    "tcp reset <tcb>",
  40.         NULLCHAR,
  41.     "rtt",        dortt,        3,    "tcp rtt <tcb> <val>",
  42.         NULLCHAR,
  43.     "status",    dotcpstat,    0,    NULLCHAR,    NULLCHAR,
  44.     "window",    dowindow,    0,    NULLCHAR,    NULLCHAR,
  45.     NULLCHAR,    NULLFP,        0,
  46.         "tcp subcommands: irtt kick mss reset rtt status window",
  47.         NULLCHAR,
  48. };
  49. int
  50. dotcp(argc,argv)
  51. int argc;
  52. char *argv[];
  53. {
  54.     return subcmd(tcpcmds,argc,argv);
  55. }
  56.  
  57. /* Eliminate a TCP connection */
  58. static int
  59. dotcpreset(argc,argv)
  60. int argc;
  61. char *argv[];
  62. {
  63.     register struct tcb *tcb;
  64.     extern char notval[];
  65.  
  66.     tcb = (struct tcb *)htol(argv[1]);
  67.     if(!tcpval(tcb)){
  68.         printf(notval);
  69.         return 1;
  70.     }
  71.     close_self(tcb,RESET);
  72.     return 0;
  73. }
  74.  
  75. /* Set initial round trip time for new connections */
  76. static int
  77. doirtt(argc,argv)
  78. int argc;
  79. char *argv[];
  80. {
  81.     if(argc < 2)
  82.         printf("%lu\n",tcp_irtt);
  83.     else
  84.         tcp_irtt = atol(argv[1]);
  85.     return 0;
  86. }
  87.  
  88. /* Set smoothed round trip time for specified TCB */
  89. static int
  90. dortt(argc,argv)
  91. int argc;
  92. char *argv[];
  93. {
  94.     register struct tcb *tcb;
  95.     extern char notval[];
  96.  
  97.     tcb = (struct tcb *)htol(argv[1]);
  98.     if(!tcpval(tcb)){
  99.         printf(notval);
  100.         return 1;
  101.     }
  102.     tcb->srtt = atol(argv[2]);
  103.     return 0;
  104. }
  105.  
  106. /* Force a retransmission */
  107. static int
  108. dotcpkick(argc,argv)
  109. int argc;
  110. char *argv[];
  111. {
  112.     register struct tcb *tcb;
  113.     extern char notval[];
  114.  
  115.     tcb = (struct tcb *)htol(argv[1]);
  116.     if(kick_tcp(tcb) == -1){
  117.         printf(notval);
  118.         return 1;
  119.     }
  120.     return 0;
  121. }
  122.  
  123. /* Set default maximum segment size */
  124. static int
  125. domss(argc,argv)
  126. int argc;
  127. char *argv[];
  128. {
  129.     if(argc < 2)
  130.         printf("%u\n",tcp_mss);
  131.     else
  132.         tcp_mss = atoi(argv[1]);
  133.     return 0;
  134. }
  135.  
  136. /* Set default window size */
  137. static int
  138. dowindow(argc,argv)
  139. int argc;
  140. char *argv[];
  141. {
  142.     if(argc < 2)
  143.         printf("%u\n",tcp_window);
  144.     else
  145.         tcp_window = atoi(argv[1]);
  146.     return 0;
  147. }
  148.  
  149. /* Display status of TCBs */
  150. static int
  151. dotcpstat(argc,argv)
  152. int argc;
  153. char *argv[];
  154. {
  155.     register struct tcb *tcb;
  156.     extern char notval[];
  157.  
  158.     if(argc < 2){
  159.         tstat();
  160.     } else {
  161.         tcb = (struct tcb *)htol(argv[1]);
  162.         if(tcpval(tcb))
  163.             state_tcp(tcb);
  164.         else
  165.             printf(notval);
  166.     }
  167.     return 0;
  168. }
  169.  
  170. /* Dump TCP stats and summary of all TCBs
  171. /*     &TCB Rcv-Q Snd-Q  Local socket           Remote socket          State
  172.  *     1234     0     0  xxx.xxx.xxx.xxx:xxxxx  xxx.xxx.xxx.xxx:xxxxx  Established
  173.  */
  174. static int
  175. tstat()
  176. {
  177.     register int i;
  178.     register struct tcb *tcb;
  179.     char *psocket();
  180.  
  181.     printf("conout %u conin %u reset out %u runt %u chksum err %u bdcsts %u\n",
  182.         tcp_stat.conout,tcp_stat.conin,tcp_stat.resets,tcp_stat.runt,
  183.         tcp_stat.checksum,tcp_stat.bdcsts);
  184. #ifdef MAC
  185.     printf("   &TCB Rcv-Q Snd-Q  Local socket           Remote socket          State\n");
  186. #else
  187.     printf("    &TCB Rcv-Q Snd-Q  Local socket           Remote socket          State\n");
  188. #endif
  189.     for(i=0;i<NTCB;i++){
  190.         for(tcb=tcbs[i];tcb != NULLTCB;tcb = tcb->next){
  191. #ifdef MAC
  192.             printf("%7lx%6u%6u  ",(long)tcb,tcb->rcvcnt,tcb->sndcnt);
  193. #else
  194.             printf("%8lx%6u%6u  ",(long)tcb,tcb->rcvcnt,tcb->sndcnt);
  195. #endif
  196.             printf("%-23s",psocket(&tcb->conn.local));
  197.             printf("%-23s",psocket(&tcb->conn.remote));
  198.             printf("%-s",tcpstates[tcb->state]);
  199.             if(tcb->state == LISTEN && (tcb->flags & CLONE))
  200.                 printf(" (S)");
  201.             printf("\n");
  202.         }
  203.     }
  204.     fflush(stdout);
  205.     return 0;
  206. }
  207. /* Dump a TCP control block in detail */
  208. static void
  209. state_tcp(tcb)
  210. struct tcb *tcb;
  211. {
  212.     int32 sent,recvd;
  213.  
  214.     if(tcb == NULLTCB)
  215.         return;
  216.     /* Compute total data sent and received; take out SYN and FIN */
  217.     sent = tcb->snd.una - tcb->iss;    /* Acknowledged data only */
  218.     recvd = tcb->rcv.nxt - tcb->irs;
  219.     switch(tcb->state){
  220.     case LISTEN:
  221.     case SYN_SENT:        /* Nothing received or acked yet */
  222.         sent = recvd = 0;    
  223.         break;
  224.     case SYN_RECEIVED:
  225.         recvd--;    /* Got SYN, no data acked yet */
  226.         sent = 0;
  227.         break;
  228.     case ESTABLISHED:    /* Got and sent SYN */
  229.     case FINWAIT1:        /* FIN not acked yet */
  230.         sent--;
  231.         recvd--;
  232.         break;
  233.     case FINWAIT2:        /* Our SYN and FIN both acked */
  234.         sent -= 2;
  235.         recvd--;
  236.         break;
  237.     case CLOSE_WAIT:    /* Got SYN and FIN, our FIN not yet acked */
  238.     case CLOSING:
  239.     case LAST_ACK:
  240.         sent--;
  241.         recvd -= 2;
  242.         break;
  243.     case TIME_WAIT:        /* Sent and received SYN/FIN, all acked */
  244.         sent -= 2;
  245.         recvd -= 2;
  246.         break;
  247.     }
  248.     printf("Local: %s",psocket(&tcb->conn.local));
  249.     printf(" Remote: %s",psocket(&tcb->conn.remote));
  250.     printf(" State: %s\n",tcpstates[tcb->state]);
  251.     printf("      Init seq    Unack     Next Resent CWind Thrsh  Wind  MSS Queue      Total\n");
  252.     printf("Send:");
  253.     printf("%9lx",tcb->iss);
  254.     printf("%9lx",tcb->snd.una);
  255.     printf("%9lx",tcb->snd.nxt);
  256.     printf("%7lu",tcb->resent);
  257.     printf("%6u",tcb->cwind);
  258.     printf("%6u",tcb->ssthresh);
  259.     printf("%6u",tcb->snd.wnd);
  260.     printf("%5u",tcb->mss);
  261.     printf("%6u",tcb->sndcnt);
  262.     printf("%11lu\n",sent);
  263.  
  264.     printf("Recv:");
  265.     printf("%9lx",tcb->irs);
  266.     printf("         ");
  267.     printf("%9lx",tcb->rcv.nxt);
  268.     printf("%7lu",tcb->rerecv);
  269.     printf("      ");
  270.     printf("      ");
  271.     printf("%6u",tcb->rcv.wnd);
  272.     printf("     ");
  273.     printf("%6u",tcb->rcvcnt);
  274.     printf("%11lu\n",recvd);
  275.  
  276.     if(tcb->reseq != (struct reseq *)NULL){
  277.         register struct reseq *rp;
  278.  
  279.         printf("Reassembly queue:\n");
  280.         for(rp = tcb->reseq;rp != (struct reseq *)NULL; rp = rp->next){
  281.             printf("  seq x%lx %u bytes\n",rp->seg.seq,rp->length);
  282.         }
  283.     }
  284.     if(tcb->backoff > 0)
  285.         printf("Backoff %u ",tcb->backoff);
  286.     if(tcb->flags & RETRAN)
  287.         printf("Retrying ");
  288.     switch(tcb->timer.state){
  289.     case TIMER_STOP:
  290.         printf("Timer stopped ");
  291.         break;
  292.     case TIMER_RUN:
  293.         printf("Timer running (%ld/%ld ms) ",
  294.          (long)MSPTICK * (tcb->timer.start - tcb->timer.count),
  295.          (long)MSPTICK * tcb->timer.start);
  296.         break;
  297.     case TIMER_EXPIRE:
  298.         printf("Timer expired ");
  299.     }
  300.     printf("SRTT %ld ms Mean dev %ld ms\n",tcb->srtt,tcb->mdev);
  301.     fflush(stdout);
  302. }
  303.  
  304.  
  305.